Written by Dare O. on 2023-06-24

Deploy a docker image to a GKE cluster

Introduction

Docker is an open platform for developing, shipping, and running containerized applications. K8s is an open source system to help automate deployment of applications and manage containerized applications. Together, these two services help aid smooth application development and deployment process.

For this short blog, I created a simple proxy Flask app to fetch random user account from Randomuser API then containerized the application and pushed docker image to docker hub.

Please note the steps below are not ideal for a PROD env as there are more steps required to prepare and run a PROD-ready Kubernetes cluster. See Here for tips for preparing a PROD-grade k8s cluster.

Prerequisites

  1. Have your application docker-image on the docker hub.

  2. A Google cloud account. Create a free account.

  3. Have kubectl installed on your local machine. Installation guide.

Creating GKE K8s cluster

To create a K8s cluster on the Google Kubernetes Engine, navigate to your Google cloud console and follow the steps below.

  • Click the Create a GKE cluster button from the console.

  • In the top right corner of the cluster creation form, click the Switch To Standard cluster and then the Create button in the next view. Note this process might take some time.

Connect to your K8s cluster

Once the cluster is ready, a green checkmark is displayed under the cluster status column. To connect to the cluster, follow the steps below.

  • Click on the cluster name.

  • At the top of the cluster info page, click on the CONNECT button. This presents a POUP window containing the gcloud container connection command. Example gcloud container clusters get-xxxxxxxxx.

  • Copy the command and execute on your favourite CLI tool. Once executed, you have an active connection to the cluster.

Create a deployment

To quickly deploy your docker image, fill in and execute the commands below.

kubectl create deployment {deployment-name} --image={docker image of docker hub}
// example kubectl create deployment flask-app --image=fancyDockerId/docker-image:tag

Expose the deployment so you can access your application.

kubectl expose deployment {deployment-name} --type=LoadBalancer --port={ ideally the port your docker image is running on}
// example kubectl expose deployment flask-app --type=LoadBalancer --port=5000

Get running service external IP address and open port by executing the command below.

kubectl get svc

// Example response: NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE flask-app LoadBalancer xx.xxx.x.20 xx.xxx.xx.118 5000:32763/TCP xxm kubernetes ClusterIP xx.xxx.x.1 443/TCP xxm

With the Cluster-IP of the LoadBalancer and it's port number, you can access your application view the web browser / Postman.

There are ways to automate the deployment process above. This can be done by creating a deployment.yaml file then simply executing a kubectl apply command. Checkout this docker guide for more on automating deployment process.

Kubectl can help you create a deployment/service.yaml configuration file. If you can have a k8s cluster running, you can exec the command below from your project's working directory.

kubectl get deployment {deployment name} -o yaml > deployment.yaml
kubectl get service {service name} -o yaml > service.yaml

// Example kubectl get deployment flask-app -o yaml > deployment.yaml kubectl get service flask-app -o yaml > service.yaml

Refresh your project's DIR to view the new .yaml configuration files. These files can now be tracked as changes within your project.

For a smoother deployment process, I will recommend merging the contents of service.yaml into deployments.yaml, so you have a single deployment file to manage your cluster. Below is what my deployment.yaml file looks like after merging service configuration and deleting some unnecessary data(Values we need not specify as they were generated by gcloud.).

apiVersion: apps/v1
kind: Deployment
metadata:
  labels:
    app: flask-app
  name: flask-app
  namespace: default
  # uid: 590128ab-8de6-4ad4-8cc6-5368a7621557
spec:
  replicas: 1
  minReadySeconds: 45
  selector:
    matchLabels:
      app: flask-app
  strategy:
    rollingUpdate:
      maxSurge: 25%
      maxUnavailable: 25%
    type: RollingUpdate
  template:
    metadata:
      labels:
        app: flask-app
    spec:
      containers:
      - image: foloo12/flask-app:0.0.1
        imagePullPolicy: IfNotPresent
        name: flask-app
        resources: {}
      restartPolicy: Always
      terminationGracePeriodSeconds: 30
---
# Service Defination
apiVersion: v1
kind: Service
metadata:
  labels:
    app: flask-app
  name: flask-app
  namespace: default
  # uid: f2f4ab6a-9841-43be-bb4d-45ef36d543e3
spec:
  ports:
  # - nodePort: 30668
  - port: 5000
    protocol: TCP
    targetPort: 5000
  selector:
    app: flask-app
  sessionAffinity: None
  type: LoadBalancer

With the deployment.yaml file, you can recreate your deployment/service or manage your cluster via the config file, so you need not manage them from Google Console thereby saving you time. As an example, I would like to scale my deployment to 2 instances, so my cluster can deal with more requests, I can simply change the spec.replicas value to 2, exec the command below and commit changes to git.

kubectl apply -f deployment.yaml

// Check if changes were applied. kubectl get rs // Number of replica set will == 2

Destroy cluster

Once you are ready to decommission services and deployments within your cluster, you can run the command below.

// Get cluster info:

kubectl get all -o wide

// Delete: kubectl delete all -l app={label for your deployment / services} // -l means label

// Example kubectl delete all -l app=flask-app